home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1999 March - Disc 1 / Macworld (1999-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / Modes / Perl Mode / perlMode.tcl < prev    next >
Encoding:
Text File  |  1998-08-13  |  10.2 KB  |  383 lines  |  [TEXT/ALFA]

  1. #############################################################################
  2. # perlMode.tcl
  3. # -----------
  4. #
  5. # This is a set of routines that allow Alpha to act as a front end for the
  6. # standalone MacPerl application and that allow Perl scripts to be used as 
  7. # text filters in Alpha.  These functions are accessed through a special 
  8. # MacPerl menu.
  9. #
  10. # The features of this package are explained in the file "MacPerl Help",
  11. # accessible from the Help menu. Version history is found in 
  12. # perlVersionHistory.tcl.
  13. #  Author: Tom Pollard
  14. #  E-mail: <pollard@schrodinger.com>
  15. #
  16. # Contributors: Dan Herron     <herron@cogsci.ucsd.edu>
  17. #               David Schooley <schooley@ee.gatech.edu>
  18. #               Vince Darley   <darley@fas.harvard.edu>
  19. #               Tom Fetherston <ranch1@earthlink.net>
  20. #               Martijn Koster <m.koster@nexor.co.uk>
  21. #
  22. #############################################################################
  23. # ◊◊◊◊ mode mini-load ◊◊◊◊ #
  24. alpha::mode Perl 3.3.2 perlMenu {*.pl *.ph *.pm} {
  25.     perlMenu electricBraces electricReturn electricSemicolon electricTab} {
  26.     addMenu perlMenu "•132"
  27.     set modeCreator(McPL) Perl
  28. #     set perlFilterMenu "textFilters"
  29. } help {file "MacPerl Help"} uninstall {this-directory}
  30.  
  31. # ◊◊◊◊ perl dummy proc's ◊◊◊◊ #
  32. proc dummyPerl {} {}
  33.  
  34. # Define the dummy proc that will be called when the perl menu
  35. # is first inserted into the menubar
  36. #
  37. proc perlMenu {} {
  38.     # had to move this from perlMenu.tcl to here to ensure newPrefs are 
  39.     # loaded before we build the menu -trf
  40.     alpha::tryToLoad "Initializing Perl menu"  perlMenu.tcl {}
  41.     #but only once
  42.     ;proc perlMenu {} {}
  43. }
  44.  
  45. #############################################################################
  46. # ◊◊◊◊ preferences ◊◊◊◊ #
  47. #  Default settings for the Perl menu flags  
  48.  
  49. newPref f perluseDebugger 0 Perl shadowPerl
  50. newPref f perlretrieveOutput 1 Perl shadowPerl
  51. newPref f perlautoSwitch 1 Perl shadowPerl
  52. newPref f perloverwriteSelection 0 Perl shadowPerl
  53. newPref f perlapplyToBuffer 1 Perl shadowPerl
  54. newPref f perlpromptForArgs 0 Perl shadowPerl
  55. newPref f perlRecycleOutput 0 Perl
  56. newPref v perlPrevScript {*startup*} Perl
  57. newPref v perlCmdlineArgs {} Perl
  58. newPref v perlVersion {5} Perl shadowPerl [list 4 5]
  59.  
  60. newPref v perlFilterPath [file join $HOME Tcl Packages "Text Filters"] Perl rebuildFilterMenu
  61. newPref v perlLibFolder "" Perl buildPerlSearchPath
  62. set Perl::commentRegexp {^[ \t]*#}
  63.  
  64. #############################################################################
  65. # Other Perl-mode variable definitions
  66.  
  67. newPref f autoMark    1    Perl
  68. newPref f wordWrap        {0} Perl
  69. newPref v funcExpr        {^[ \t]*sub\s+[_\w:]+(\s+\([$@%*;\]+\))?\s*\{} Perl
  70. newPref v prefixString    {# } Perl
  71. newPref v wordBreak        {(([$%@*]?[_\w]+)|(\$?[][&_`'+*./|,\\";#%=\~^:?!@\$<>()-])|((\$\^)\w))}
  72. newPref v wordBreakPreface        {([^a-zA-Z0-9%_@*\$^]|.\$)} Perl
  73. newPref v stringColor    green    Perl
  74.  
  75. # ALL THE ABOVE VARS ARE NOW GLOBAL AND MODE-VARS
  76. # unsetting old prefs variables
  77.  
  78. catch {unset PerlmodeVars(elecLBrace)}
  79. catch {unset PerlmodeVars(elecRBrace)}
  80. catch {unset PerlmodeVars(electricReturn)}
  81. catch {unset PerlmodeVars(electricSemi)}
  82. catch {unset PerlmodeVars(electricTab)}
  83.  
  84. #############################################################################
  85. # ◊◊◊◊ paths to standard files ◊◊◊◊ #
  86. #  Return paths to standard files, based on the path to MacPerl:
  87. #
  88. proc macperlFolder {} {
  89.     return [file dirname [nameFromAppl McPL]]
  90. }
  91.  
  92. proc stdinPath {} {
  93.    return [file join [macperlFolder] STDIN]
  94. }
  95.  
  96. proc scriptPath {} {
  97.    return [file join [macperlFolder] SCRIPT]
  98. }
  99.  
  100. # ◊◊◊◊ dividers (code sectioning) ◊◊◊◊ #
  101.  
  102. ## 
  103.  # -------------------------------------------------------------------------
  104.  # 
  105.  # "Perl::insertDivider" --
  106.  # 
  107.  #  Modified from Vince's original to allow you to just select part of
  108.  #  an already written comment and turn it into a Divider. -trf
  109.  # -------------------------------------------------------------------------
  110.  ##
  111. proc Perl::insertDivider {} {
  112.     if {[isSelection]} {
  113.         set enfoldThis [getSelect]
  114.         beginningOfLine
  115.         killLine
  116.         insertText "##### $enfoldThis #####"
  117.         return
  118.     } 
  119.     elec::Insertion "##### •• #####"
  120. }
  121. Bind 0x14 <z> Perl::insertDivider Perl
  122.  
  123.  
  124. #############################################################################
  125. # ◊◊◊◊ Marking ◊◊◊◊ #
  126.  
  127. ##############################################################################
  128. # Automatic indexing of Perl subs
  129. #
  130. # called by the "M" button     Modified -trf
  131. proc Perl::parseFuncs {} {
  132.     set end [maxPos]
  133.     set pos [minPos]
  134.     set l {}
  135.     set markExpr {^[ \t]*sub\s+[_\w:]+(\s+\([$@%*;\]+\))?\s*\{}
  136.     set appearanceList {}
  137.     while {![catch {search -s -f 1 -r 1 -m 0 -i 0 "$markExpr" $pos} res]} {
  138.         set start [lindex $res 0]
  139.         set end [lindex $res 1]
  140.         set t [eval getText $res]
  141.         
  142.         switch -regexp $t {
  143.             "sub" {
  144.                 regexp {^([ \t]*)sub\s+([_\w:]+)(\s+\(([$@%*;\]+)\))?\s*\{} $t all indent subName argTypes
  145.                 set word $subName 
  146.             }
  147.         }
  148.         if {$argTypes != {}} {
  149.             set argLabel "$word$argTypes" 
  150.         } else {
  151.             set argLabel $word
  152.         } 
  153.         if {[info exists cnts($word)]} {
  154.             # This section handles duplicate. i.e., overloaded names
  155.             set cnts($word) [expr $cnts($word) + 1]
  156.             set tailOfTag($word) " (1 of $cnts($word))"
  157.         } else {
  158.             #SO do: remember the following
  159.             set cnts($word) 1
  160.             # if this is the only occurence of this proc, remember where it starts
  161.             set indx($word) [lineStart [expr $start - 1]]
  162.         }
  163.         #associate name and tag
  164.         set tag($word) $argLabel
  165.         
  166.         #advance pos to where we want to start the next search from
  167.         set pos $end
  168.     }
  169.  
  170.     set rtnRes {}
  171.     
  172.     if {[info exists indx]} {
  173.         foreach hn [lsort -ignore [array names indx]] {
  174.             set next [nextLineStart $indx($hn)]
  175.             set completeTag [set tag($hn)]
  176.             if {[info exists tailOfTag($hn)]} {
  177.                 append completeTag [ set tailOfTag($hn) ]
  178.             }
  179.             
  180.             lappend rtnRes $completeTag $next
  181.         }
  182.     }
  183.     return $rtnRes 
  184. }
  185.  
  186.  
  187. proc Perl::MarkFile {} {
  188.     global PerlmodeVars
  189.     
  190.     # this is a global var in tcl where this was taken from
  191.     set structuralMarks 1
  192.     set pos [minPos] ;#pos to start/continue search
  193.     set l {} 
  194.     set asEncountered {}
  195.     
  196.     #With this regex we scan for 
  197.     # a package followed by a block with indented sub's
  198.     # a package statement with just normal, non-indented sub's
  199.     # {
  200.     #     (
  201.     #     ^
  202.     #     (
  203. #              package\s+[_\w:]+\s*;\s*\{
  204. #              |package\s+[_\w:]+\s*;
  205. #              |BEGIN
  206. #              |END
  207. #              |sub\s+[_\w:]+(\s+\([$@%*;\]+\))?\s*\{
  208. #              |[ \t]+sub\s+[_\w:]+(\s+\([$@%*;\]+\))?\s*\{
  209. #              |=head1
  210. #              |=head2(.*)Section
  211. #              |=pod
  212. #              |__END__
  213. #              |__DATA__
  214.     #     )
  215.     #     )
  216.     # }
  217.     # 
  218.     # #
  219. #     set markExpr {(^(package\s+[_\w:]+\s*;\s*\{|BEGIN|END|sub\s+[_\w:]+(\s+\([$@%*;\]+\))?\s*\{|=head1|=pod|__END__|__DATA__)(\s+[^\s;\{])*)}
  220.     set markExpr {^(package\s+[_\w:]+\s*;\s*\{*|BEGIN|END|[ \t]*sub\s+[_\w:]+(\s+\([$@%*;\]+\))?\s*\{|=head1|=pod|__END__|__DATA__)}
  221.     set pos 0
  222.     set l {}
  223.     if $structuralMarks {
  224.         append markExpr {|(^ *###+ ([^#]+) ###+)}
  225.     } 
  226.     
  227.     set hasMarkers 0
  228.     set inPackageSep  {} 
  229.     set allowIndentedSubs 0
  230.     set pkgBlockEndPos 0
  231.     
  232.     while {![catch {search -s -f 1 -r 1 -m 0 -i 0 "$markExpr" $pos} res]} {
  233.         set start [lindex $res 0]
  234.         set end [lindex $res 1]
  235.         set t [eval getText $res]
  236.         
  237.         switch -regexp $t {
  238.             "^package" {
  239.                 regexp {^package\s+([_\w:]+)\s*;\s*(\{)*} $t all text blockBeg
  240.                 if {[set blockBeg] !=  {} } {
  241. #                     #we have seen a "block-like package"
  242. #                     set allowIndentedSubs 1
  243.                     #determine where "package block" ends
  244.                     set pkgBlockEndPos [matchIt "\{" [expr $end + 1]]
  245.                     #
  246.                 } 
  247.                 if {$structuralMarks} {
  248.                     set text "$text •pkg"
  249.                     set inPackageSep "::"
  250.                 } else {
  251.                     set pos $end
  252.                     continue
  253.                 }         
  254.             }
  255.             "BEGIN" {
  256.                 set text " BEGIN"
  257.             }             
  258.             "^END" {
  259.                 set text " END"
  260.             }             
  261.             {sub\s+[_\w:]+;} {
  262.                  set pos $end
  263.                 continue
  264.             }
  265.             {^[ \t]+sub} {
  266.                 if {[set start] >= [set pkgBlockEndPos]} {
  267.                      set pos $end
  268.                     continue
  269.                 } 
  270.                 regexp {^(([ \t]*)sub\s+)([\w_:]+)} $t all preNameText indent text
  271.                 if {$structuralMarks} {
  272.                     set text " $inPackageSep$text"
  273.                     set start [lineStart [expr $start + [string length $preNameText] + 1]]
  274.                 }
  275.             }
  276.             "^sub" {
  277.                 regexp {^(sub\s+)([\w_:]+)} $t all preNameText text
  278.                 if {$structuralMarks} {
  279.                     set text " $inPackageSep$text"
  280.                     set start [lineStart [expr $start + [string length $preNameText] + 1]]
  281.                 }             
  282.             }
  283.             "###+" { 
  284.                 regexp {###+ ([^#]+) ###+} $t all text
  285.                 if {[regexp "^(    )|(    )###+" $t]} {
  286.                     set text " •$text"
  287.                 } else {
  288.                     set text "•$text"
  289.                 }     
  290.                 set hasMarkers 1
  291.             }
  292.             "=head1" -
  293.             "=pod" {
  294.                 set pos $end
  295.                 if {![catch {search -s -f 1 -r 1 -m 0 -i 0 "^=cut" $pos} res]} {
  296.                     set start [lindex $res 0]
  297.                     set end [nextLineStart $start]
  298.                     continue
  299.                 } else {
  300.                     message "*warning* - embeded pod with no cut encountered"
  301.                     break
  302.                 } 
  303.             }             
  304.             "__END__" -
  305.             "__DATA__" {
  306.                 break
  307.             }             
  308.             "default" {
  309.                 set text ""
  310.                 continue
  311.             }             
  312.         }
  313.         set pos $end
  314.         
  315.         if {$structuralMarks} {
  316.             while { [lsearch -exact $asEncountered $text] != -1 } {
  317.                 set text "$text "
  318.             }
  319.             lappend asEncountered $text
  320.             set arr inds
  321.         } 
  322.         set ${arr}($text) $start
  323.     }
  324.  
  325.     set already ""
  326.     foreach arr {inds} {
  327.         if {[info exists $arr]} {
  328.             if $structuralMarks {
  329.                 set order $asEncountered
  330.             } 
  331.             foreach f $order {
  332.                 set el [set ${arr}($f)]
  333.                 set ff $f
  334.                 while { [lsearch -exact $already $ff] != -1 } {
  335.                     set ff "$ff "
  336.                 }
  337.                 lappend already $ff
  338.                 if {$hasMarkers && ![string match "•*" $ff] } {
  339.                     set ff " $ff"
  340.                 } 
  341.                 setNamedMark $ff $el $el $el
  342.             }
  343.         }
  344.     }
  345. }
  346.  
  347.  
  348. # ◊◊◊◊ electric behaviour ◊◊◊◊ #
  349. proc Perl::electricLeft {} {
  350.     set prevChar [lookAt [pos::math [getPos] - 1]]
  351.     if {$prevChar == " " || $prevChar == "\)"} {
  352.     ::electricLeft
  353.     return
  354.     }
  355.     deleteText [getPos] [selEnd]
  356.     insertText "\{"
  357. }
  358.  
  359. proc Perl::electricRight {} {
  360.     set prevChar [lookAt [pos::math [getPos] - 1]]
  361.     if {$prevChar == " " || $prevChar == ";" || $prevChar == "\t" || $prevChar == "\}"} {
  362.     ::electricRight
  363.     return
  364.     }
  365.     deleteText [getPos] [selEnd]
  366.     insertText "\}"
  367.     catch {blink [matchIt "\}" [pos::math [getPos] - 2]]}
  368.     return
  369. }
  370.  
  371. # ◊◊◊◊ Inintialize Perl mode ◊◊◊◊ #
  372. if ![alpha::tryToLoad "Initializing Perl" \
  373.   "perl$PerlmodeVars(perlVersion).tcl" {}\
  374.   perlEngine.tcl {}\
  375.   perlFilters&Misc.tcl {}] {
  376.     alertnote "Error: Not all of the mode files loaded"
  377. }
  378.  
  379.  
  380.  
  381.